重写overriding
视频锁定
{$ currentTime | date:'mm:ss' $}
{$ timeLeft | date:'mm:ss' $}
重写(overriding)是子类对父类允许访问的方法进行重新编写,其返回值和参数都不能改变(即方法签名不能改变)。重写的好处在于子类可以根据需要定义自己的行为,也就是说子类能够根据需要重新实现父类的方法。在面向对象原则里,重写意味着可以重写任何现有方法。代码示例如下:
class Animal {
public void move() {
System.out.println("动物可以行走");
}
}
class Dog extends Animal {
public void move() {
System.out.println("狗可以行走");
}
}
public class TestDog {
public static void main(String args[]) {
Animal a = new Animal(); //创建Animal对象
Animal b = new Dog(); //创建Dog对象
a.move(); //执行Animal的move()方法
b.move(); //执行Dog的move()方法
}
}
运行结果如下:
动物可以行走
狗可以行走
在上面的例子中可以看到,尽管b属于Animal类型,但是它运行的是Dog类的move()
方法。
这是由于在编译阶段,只是检查参数的引用类型。
然而在运行时,Java虚拟机(JVM)指定对象的类型并且运行该对象的方法。
因此在上面的例子中,之所以能编译成功,是因为Animal类中存在move()
方法,然而运行时,运行的是特定对象的方法。
思考以下例子:
class Animal {
public void move() {
System.out.println("动物可以行走");
}
}
class Dog extends Animal {
public void move() {
System.out.println("狗可以行走");
}
public void bark() {
System.out.println("狗可以叫");
}
}
public class TestDog {
public static void main(String args[]) {
Animal a = new Animal(); //创建Animal对象
Animal b = new Dog(); //创建Dog对象
a.move(); //执行Animal类的move()方法
b.move(); //执行Dog类的move()方法
b.bark();
}
}
运行结果如下:
TestDog.java:30: cannot find symbol
symbol : method bark()
location: class Animal
b.bark();
^
该程序将抛出一个编译错误,因为b
的引用类型Animal没有bark()
方法。
重写的规则:
- 参数列表必须完全与被重写方法的相同;
- 返回类型必须完全与被重写方法的返回类型相同;
- 访问权限不能比父类中被重写的方法的访问权限更高。例如:如果父类的一个方法被声明为public,那么在子类中重写该方法就不能声明为protected。
- 父类的成员方法只能被它的子类重写。
- 声明为final的方法不能被重写。
- 声明为static的方法不能被重写,但是能够被再次声明。
- 如果一个方法不能被继承,那么该方法不能被重写。
- 子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为private和final的方法。
- 子类和父类不在同一个包中,那么子类只能够重写父类的声明为public和protected的非final方法。
- 重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。
- 构造方法不能被重写。
- 如果不能继承一个方法,则不能重写这个方法。
在线练习
{$ activeFileHint $}